## COMS12200 lab. worksheet: week #2

- We intend this worksheet to be attempted in the associated lab. session, which represents a central form of help and feedback for this unit.
- The worksheet is not *directly* assessed. Rather, it simply provides guided, practical exploration of the material covered in the lecture(s): the only requirement is that you archive your work (complete or otherwise) in a portfolio via the appropriate component at

https://wwwa.fen.bris.ac.uk/COMS12200/

*This* forms the basis for assessment during a viva at the end of each teaching block, by acting as evidence that you have engaged with and understand the material.

- The deadline for submission to your portfolio is the end of the associated teaching block (i.e., in time for the viva): there is no requirement to complete the worksheet in the lab. itself (some questions require too much work to do so), but there is an emphasis on *you* to catch up with anything left incomplete.
- To accommodate the number of students registered on the unit, the single 3 hour timetabled lab. session is split into two  $1\frac{1}{2}$  hour halves. You should attend *one* half only, selecting as follows:
  - 1. if you have a timetable clash that means you *must* attend one half or the other then do so, otherwise
  - 2. execute the following BASH command pipeline

e.g., log into a lab. workstation and copy-and-paste it into a terminal window, then check the output: 0 means attend the first half, 1 means attend the second half.

**S1.** There is a set of solutions available at

http://www.cs.bris.ac.uk/home/page/teaching/material/arch\_old/sheet/exam-prelims\_boolean.s.pdf

**S2.** This is just a simple starting point designed to ensure you are comfortable with features on the board. The idea is just to reproduce the truth table for NAND, i.e.,

| y | $x \wedge y$ |
|---|--------------|
| 0 | 1            |
| 1 | 1            |
| 0 | 1            |
| 1 | 0            |
|   | 0            |

making sure the output in each row matches the physical computation on the board when the input pins are connected appropriately.

S3. The first three cases should all be easy to replicate because a) they only need a few NAND operators, and b) were covered in the lecture(s). Specifically, we saw the following identities:

$$\begin{array}{rcl} \neg x & \equiv & x \,\overline{\wedge}\, x \\ x \wedge y & \equiv & (x \,\overline{\wedge}\, y) & \overline{\wedge} & (x \,\overline{\wedge}\, y) \\ x \vee y & \equiv & (x \,\overline{\wedge}\, x) & \overline{\wedge} & (y \,\overline{\wedge}\, y) \end{array}$$

Note that to compute  $x \wedge x$ , you need two jumper wires connected to the NAND operator input pins; both need either to be disconnected (meaning x = 0) or connected to 1, because both represent the same x. This is a little awkward, so it might be easier to use the fact that

$$\neg x \equiv x \wedge 1.$$

Now the second input pin is fixed to 1, meaning you only have one jumper wire representing x (more closely matching the NOT operator).

The case of XOR is more difficult. We saw in the lecture(s) that it can be written in various different ways, e.g.,

$$x \oplus y \equiv (\neg x \land y) \lor (x \land \neg y)$$
  
$$\equiv (x \lor y) \land \neg (x \land y)$$

For the sake of argument imagine we instead opt for the former: just by applying the identities above, we can rewrite it as

so that  $x \oplus y \equiv t_8$  having used 9 operators. We can then make incremental improvements by inspection.  $t_3$  computes  $\neg t_2 = t_2 \land t_2$  for example, and then later  $t_6$  computes  $\neg t_3$  in the same way. So basically, since  $t_6 = \neg \neg t_2$ , we can eliminate both NOTs via the involution axiom, and similarly for  $t_7$ , to get

$$\begin{array}{rclcrcl} t_0 & = & x & \overline{\wedge} & x \\ t_1 & = & y & \overline{\wedge} & y \\ t_2 & = & t_0 & \overline{\wedge} & y \\ t_4 & = & t_1 & \overline{\wedge} & x \\ t_8 & = & t_2 & \overline{\wedge} & t_4 \end{array}$$

having now used only 5 operators: this *seems* to be about the best we can hope for. But imagine we opt for the latter expression that describes XOR instead of the former. Some initial manipulation allows us to rewrite it as follows

$$x \oplus y \equiv (x \lor y) \land \neg(x \land y)$$

$$\equiv \neg(x \land y) \land (x \lor y) \qquad \text{(commutativity)}$$

$$\equiv (x \land \neg(x \land y)) \lor (y \land \neg(x \land y)) \qquad \text{(distribution)}$$

$$\equiv \neg \neg((x \land \neg(x \land y)) \lor (y \land \neg(x \land y))) \qquad \text{(involution)}$$

$$\equiv \neg(\neg(x \land \neg(x \land y)) \land \neg(y \land \neg(x \land y))) \qquad \text{(de Morgan)}$$

$$\equiv (x \overleftarrow{\land} (x \overleftarrow{\land} y)) \overleftarrow{\land} (y \overleftarrow{\land} (x \overleftarrow{\land} y))$$

It might seem odd to call this simplification, because the result may look more complicated! Crucially however, *every* (sub-)term is now computed using NAND: the expression can therefore be rewritten as

so that  $x \oplus y \equiv t_3$  now using just 4 operators.

Given an expression in SoP (resp. PoS) form, the example above yields a fairly general strategy for simplification ready for implementation using NAND (resp. NOR) alone. In short, introducing what *seem* to be redundant NOT operators turns out to be an advantage, because we can then apply the de Morgan axiom: this "pushes" a NOT into the expression, swapping ANDs into NANDs etc.

b The first step is to formulate a working design. We want to produce the following behaviour, as translated from the description:

| а | b | С | Maj(a,b,c) |
|---|---|---|------------|
| 0 | 0 | 0 | 0          |
| 0 | 0 | 1 | 0          |
| 0 | 1 | 0 | 0          |
| 0 | 1 | 1 | 1          |
| 1 | 0 | 0 | 0          |
| 1 | 0 | 1 | 1          |
| 1 | 1 | 0 | 1          |
| 1 | 1 | 1 | 1          |

Note that if three of the inputs are 1 then obviously two of them are 1, so it suffices to detect the latter case only. This reduces the amount of work required, because we already know AND will produce 1 as an output if both inputs are 1: we can use AND to "detect" each of three cases where two or more inputs are 1. Put another way, we can express the function as

$$Maj(a, b, c) = (b \wedge c) \vee (a \wedge c) \vee (a \wedge b),$$

i.e., "the output is 1 when either b = 1 and c = 1, or a = 1 and c = 1, or a = 1 and b = 1".

The next step is to translate this into NAND operators. Conceptually this is easy, because we just apply the known identities (as we did above with XOR). However, this quickly becomes hard to manage: when written out naively in full, we use around 39 NAND operators. With more care, we can share numerous common sub-terms and write

where  $Maj(a, c, b) = t_{11}$  having used 12 operators. But we can again make improvements to this with initial simplification of our expression. Applying roughly the same strategy as with XOR, we find that

$$\begin{aligned} \mathrm{Maj}(a,b,c) &=& (b \wedge c) \vee (a \wedge c) \vee (a \wedge b) \\ &\equiv & (b \wedge c) \vee (a \wedge (c \vee b)) & (\mathrm{distribution}) \\ &\equiv & \neg \neg ((b \wedge c) \vee (a \wedge (c \vee b))) & (\mathrm{involution}) \\ &\equiv & \neg (\neg (b \wedge c) \wedge \neg (a \wedge (c \vee b))) & (\mathrm{de\ Morgan}) \\ &\equiv & \neg (\neg (b \wedge c) \wedge \neg (a \wedge \neg \neg (c \vee b))) & (\mathrm{involution}) \\ &\equiv & \neg (\neg (b \wedge c) \wedge \neg (a \wedge \neg (\neg c \wedge \neg b))) & (\mathrm{de\ Morgan}) \\ &\equiv & (b \overline{\wedge} c) \overline{\wedge} (a \overline{\wedge} ((c \overline{\wedge} c) \overline{\wedge} (b \overline{\wedge} b))) & \end{aligned}$$

and produce an implementation

$$\begin{array}{rclcrcl} t_0 & = & b & \overline{\wedge} & c \\ t_1 & = & c & \overline{\wedge} & c \\ t_2 & = & b & \overline{\wedge} & b \\ t_3 & = & t_1 & \overline{\wedge} & t_2 \\ t_4 & = & a & \overline{\wedge} & t_3 \\ t_5 & = & t_0 & \overline{\wedge} & t_4 \end{array}$$

where  $Maj(a, c, b) = t_5$  now using just 6 operators.

**S4[+].** To start with, think a bit about what the majority function is doing: you can think of it as an exercise in voting, with the output representing a majority decision between voters *a*, *b*, and *c*. For instance if two or more vote 1 then the output is 1; otherwise two or more must have voted 0 meaning the output is 0.

By using the majority function as suggested in the question, you produce a component called a **C-element**. Now there are only two voters, but also a "default result" that decides the output in case of a tie. Put another way, we can write the as

| а | b | $c' = \mathrm{Maj}(a, b, c)$ |
|---|---|------------------------------|
| 0 | 0 | 0                            |
| 0 | 1 | c                            |
| 1 | 0 | c                            |
| 1 | 1 | 1                            |

where c and c' are the values of c before and after we set a and b. Put another way, the middle two rows are saying that if the value was c = 0 (resp. c = 1), then it will *remain* c' = 0 (resp. c' = 1); in contrast, the top and bottom rows are saying that c' is *forced* to 0 and 1 respectively, regardless of what c was before.

In short, this component retains or remembers some state over time; this differs significantly from those previously seen, the majority function for example, where the output changes (and is lost) as soon as the inputs change. Since *c* is representing 1 bit, the C-element acts as a cell that can store 1-bit values, very roughly like a memory cell does. In a later lecture(s), we see that it falls within a larger class of components called latches.